merge.js ➔ cloneHunk   A
last analyzed

Complexity

Conditions 1
Paths 1

Size

Total Lines 7
Code Lines 5

Duplication

Lines 0
Ratio 0 %

Importance

Changes 0
Metric Value
cc 1
eloc 5
nc 1
nop 2
dl 0
loc 7
rs 10
c 0
b 0
f 0
1
/*istanbul ignore start*/'use strict';
2
3
exports.__esModule = true;
4
exports. /*istanbul ignore end*/calcLineCount = calcLineCount;
5
/*istanbul ignore start*/exports. /*istanbul ignore end*/merge = merge;
6
7
var /*istanbul ignore start*/_create = require('./create') /*istanbul ignore end*/;
8
9
var /*istanbul ignore start*/_parse = require('./parse') /*istanbul ignore end*/;
10
11
var /*istanbul ignore start*/_array = require('../util/array') /*istanbul ignore end*/;
12
13
/*istanbul ignore start*/function _toConsumableArray(arr) { if (Array.isArray(arr)) { for (var i = 0, arr2 = Array(arr.length); i < arr.length; i++) { arr2[i] = arr[i]; } return arr2; } else { return Array.from(arr); } }
0 ignored issues
show
Comprehensibility introduced by
else is not necessary here since all if branches return, consider removing it to reduce nesting and make code more readable.
Loading history...
14
15 View Code Duplication
/*istanbul ignore end*/function calcLineCount(hunk) {
0 ignored issues
show
Duplication introduced by
This code seems to be duplicated in your project.
Loading history...
16
  var conflicted = false;
17
18
  hunk.oldLines = 0;
19
  hunk.newLines = 0;
20
21
  hunk.lines.forEach(function (line) {
22
    if (typeof line !== 'string') {
23
      conflicted = true;
24
      return;
25
    }
26
27
    if (line[0] === '+' || line[0] === ' ') {
28
      hunk.newLines++;
29
    }
30
    if (line[0] === '-' || line[0] === ' ') {
31
      hunk.oldLines++;
32
    }
33
  });
34
35
  if (conflicted) {
36
    delete hunk.oldLines;
37
    delete hunk.newLines;
38
  }
39
}
40
41 View Code Duplication
function merge(mine, theirs, base) {
0 ignored issues
show
Duplication introduced by
This code seems to be duplicated in your project.
Loading history...
42
  mine = loadPatch(mine, base);
43
  theirs = loadPatch(theirs, base);
44
45
  var ret = {};
46
47
  // For index we just let it pass through as it doesn't have any necessary meaning.
48
  // Leaving sanity checks on this to the API consumer that may know more about the
49
  // meaning in their own context.
50
  if (mine.index || theirs.index) {
51
    ret.index = mine.index || theirs.index;
52
  }
53
54
  if (mine.newFileName || theirs.newFileName) {
55
    if (!fileNameChanged(mine)) {
56
      // No header or no change in ours, use theirs (and ours if theirs does not exist)
57
      ret.oldFileName = theirs.oldFileName || mine.oldFileName;
58
      ret.newFileName = theirs.newFileName || mine.newFileName;
59
      ret.oldHeader = theirs.oldHeader || mine.oldHeader;
60
      ret.newHeader = theirs.newHeader || mine.newHeader;
61
    } else if (!fileNameChanged(theirs)) {
62
      // No header or no change in theirs, use ours
63
      ret.oldFileName = mine.oldFileName;
64
      ret.newFileName = mine.newFileName;
65
      ret.oldHeader = mine.oldHeader;
66
      ret.newHeader = mine.newHeader;
67
    } else {
68
      // Both changed... figure it out
69
      ret.oldFileName = selectField(ret, mine.oldFileName, theirs.oldFileName);
70
      ret.newFileName = selectField(ret, mine.newFileName, theirs.newFileName);
71
      ret.oldHeader = selectField(ret, mine.oldHeader, theirs.oldHeader);
72
      ret.newHeader = selectField(ret, mine.newHeader, theirs.newHeader);
73
    }
74
  }
75
76
  ret.hunks = [];
77
78
  var mineIndex = 0,
79
      theirsIndex = 0,
80
      mineOffset = 0,
81
      theirsOffset = 0;
82
83
  while (mineIndex < mine.hunks.length || theirsIndex < theirs.hunks.length) {
84
    var mineCurrent = mine.hunks[mineIndex] || { oldStart: Infinity },
85
        theirsCurrent = theirs.hunks[theirsIndex] || { oldStart: Infinity };
86
87
    if (hunkBefore(mineCurrent, theirsCurrent)) {
88
      // This patch does not overlap with any of the others, yay.
89
      ret.hunks.push(cloneHunk(mineCurrent, mineOffset));
90
      mineIndex++;
91
      theirsOffset += mineCurrent.newLines - mineCurrent.oldLines;
92
    } else if (hunkBefore(theirsCurrent, mineCurrent)) {
93
      // This patch does not overlap with any of the others, yay.
94
      ret.hunks.push(cloneHunk(theirsCurrent, theirsOffset));
95
      theirsIndex++;
96
      mineOffset += theirsCurrent.newLines - theirsCurrent.oldLines;
97
    } else {
98
      // Overlap, merge as best we can
99
      var mergedHunk = {
100
        oldStart: Math.min(mineCurrent.oldStart, theirsCurrent.oldStart),
101
        oldLines: 0,
102
        newStart: Math.min(mineCurrent.newStart + mineOffset, theirsCurrent.oldStart + theirsOffset),
103
        newLines: 0,
104
        lines: []
105
      };
106
      mergeLines(mergedHunk, mineCurrent.oldStart, mineCurrent.lines, theirsCurrent.oldStart, theirsCurrent.lines);
107
      theirsIndex++;
108
      mineIndex++;
109
110
      ret.hunks.push(mergedHunk);
111
    }
112
  }
113
114
  return ret;
115
}
116
117
function loadPatch(param, base) {
118
  if (typeof param === 'string') {
119
    if (/^@@/m.test(param) || /^Index:/m.test(param)) {
120
      return (/*istanbul ignore start*/(0, _parse.parsePatch) /*istanbul ignore end*/(param)[0]
0 ignored issues
show
Comprehensibility introduced by
Usage of the sequence operator is discouraged, since it may lead to obfuscated code.

The sequence or comma operator allows the inclusion of multiple expressions where only is permitted. The result of the sequence is the value of the last expression.

This operator is most often used in for statements.

Used in another places it can make code hard to read, especially when people do not realize it even exists as a seperate operator.

This check looks for usage of the sequence operator in locations where it is not necessary and could be replaced by a series of expressions or statements.

var a,b,c;

a = 1, b = 1,  c= 3;

could just as well be written as:

var a,b,c;

a = 1;
b = 1;
c = 3;

To learn more about the sequence operator, please refer to the MDN.

Loading history...
121
      );
122
    }
123
124
    if (!base) {
125
      throw new Error('Must provide a base reference or pass in a patch');
126
    }
127
    return (/*istanbul ignore start*/(0, _create.structuredPatch) /*istanbul ignore end*/(undefined, undefined, base, param)
0 ignored issues
show
Comprehensibility introduced by
Usage of the sequence operator is discouraged, since it may lead to obfuscated code.

The sequence or comma operator allows the inclusion of multiple expressions where only is permitted. The result of the sequence is the value of the last expression.

This operator is most often used in for statements.

Used in another places it can make code hard to read, especially when people do not realize it even exists as a seperate operator.

This check looks for usage of the sequence operator in locations where it is not necessary and could be replaced by a series of expressions or statements.

var a,b,c;

a = 1, b = 1,  c= 3;

could just as well be written as:

var a,b,c;

a = 1;
b = 1;
c = 3;

To learn more about the sequence operator, please refer to the MDN.

Loading history...
128
    );
129
  }
130
131
  return param;
132
}
133
134
function fileNameChanged(patch) {
135
  return patch.newFileName && patch.newFileName !== patch.oldFileName;
136
}
137
138
function selectField(index, mine, theirs) {
139
  if (mine === theirs) {
140
    return mine;
141
  } else {
0 ignored issues
show
Comprehensibility introduced by
else is not necessary here since all if branches return, consider removing it to reduce nesting and make code more readable.
Loading history...
142
    index.conflict = true;
143
    return { mine: mine, theirs: theirs };
144
  }
145
}
146
147
function hunkBefore(test, check) {
148
  return test.oldStart < check.oldStart && test.oldStart + test.oldLines < check.oldStart;
149
}
150
151
function cloneHunk(hunk, offset) {
152
  return {
153
    oldStart: hunk.oldStart, oldLines: hunk.oldLines,
154
    newStart: hunk.newStart + offset, newLines: hunk.newLines,
155
    lines: hunk.lines
156
  };
157
}
158
159 View Code Duplication
function mergeLines(hunk, mineOffset, mineLines, theirOffset, theirLines) {
0 ignored issues
show
Duplication introduced by
This code seems to be duplicated in your project.
Loading history...
160
  // This will generally result in a conflicted hunk, but there are cases where the context
161
  // is the only overlap where we can successfully merge the content here.
162
  var mine = { offset: mineOffset, lines: mineLines, index: 0 },
163
      their = { offset: theirOffset, lines: theirLines, index: 0 };
164
165
  // Handle any leading content
166
  insertLeading(hunk, mine, their);
167
  insertLeading(hunk, their, mine);
168
169
  // Now in the overlap content. Scan through and select the best changes from each.
170
  while (mine.index < mine.lines.length && their.index < their.lines.length) {
171
    var mineCurrent = mine.lines[mine.index],
172
        theirCurrent = their.lines[their.index];
173
174
    if ((mineCurrent[0] === '-' || mineCurrent[0] === '+') && (theirCurrent[0] === '-' || theirCurrent[0] === '+')) {
175
      // Both modified ...
176
      mutualChange(hunk, mine, their);
177
    } else if (mineCurrent[0] === '+' && theirCurrent[0] === ' ') {
178
      /*istanbul ignore start*/var _hunk$lines;
179
180
      /*istanbul ignore end*/ // Mine inserted
181
      /*istanbul ignore start*/(_hunk$lines = /*istanbul ignore end*/hunk.lines).push. /*istanbul ignore start*/apply /*istanbul ignore end*/( /*istanbul ignore start*/_hunk$lines /*istanbul ignore end*/, /*istanbul ignore start*/_toConsumableArray( /*istanbul ignore end*/collectChange(mine)));
182
    } else if (theirCurrent[0] === '+' && mineCurrent[0] === ' ') {
183
      /*istanbul ignore start*/var _hunk$lines2;
184
185
      /*istanbul ignore end*/ // Theirs inserted
186
      /*istanbul ignore start*/(_hunk$lines2 = /*istanbul ignore end*/hunk.lines).push. /*istanbul ignore start*/apply /*istanbul ignore end*/( /*istanbul ignore start*/_hunk$lines2 /*istanbul ignore end*/, /*istanbul ignore start*/_toConsumableArray( /*istanbul ignore end*/collectChange(their)));
187
    } else if (mineCurrent[0] === '-' && theirCurrent[0] === ' ') {
188
      // Mine removed or edited
189
      removal(hunk, mine, their);
190
    } else if (theirCurrent[0] === '-' && mineCurrent[0] === ' ') {
191
      // Their removed or edited
192
      removal(hunk, their, mine, true);
193
    } else if (mineCurrent === theirCurrent) {
194
      // Context identity
195
      hunk.lines.push(mineCurrent);
196
      mine.index++;
197
      their.index++;
198
    } else {
199
      // Context mismatch
200
      conflict(hunk, collectChange(mine), collectChange(their));
201
    }
202
  }
203
204
  // Now push anything that may be remaining
205
  insertTrailing(hunk, mine);
206
  insertTrailing(hunk, their);
207
208
  calcLineCount(hunk);
209
}
210
211 View Code Duplication
function mutualChange(hunk, mine, their) {
0 ignored issues
show
Duplication introduced by
This code seems to be duplicated in your project.
Loading history...
212
  var myChanges = collectChange(mine),
213
      theirChanges = collectChange(their);
214
215
  if (allRemoves(myChanges) && allRemoves(theirChanges)) {
216
    // Special case for remove changes that are supersets of one another
217
    if ( /*istanbul ignore start*/(0, _array.arrayStartsWith) /*istanbul ignore end*/(myChanges, theirChanges) && skipRemoveSuperset(their, myChanges, myChanges.length - theirChanges.length)) {
0 ignored issues
show
Comprehensibility introduced by
Usage of the sequence operator is discouraged, since it may lead to obfuscated code.

The sequence or comma operator allows the inclusion of multiple expressions where only is permitted. The result of the sequence is the value of the last expression.

This operator is most often used in for statements.

Used in another places it can make code hard to read, especially when people do not realize it even exists as a seperate operator.

This check looks for usage of the sequence operator in locations where it is not necessary and could be replaced by a series of expressions or statements.

var a,b,c;

a = 1, b = 1,  c= 3;

could just as well be written as:

var a,b,c;

a = 1;
b = 1;
c = 3;

To learn more about the sequence operator, please refer to the MDN.

Loading history...
218
      /*istanbul ignore start*/var _hunk$lines3;
219
220
      /*istanbul ignore end*/ /*istanbul ignore start*/(_hunk$lines3 = /*istanbul ignore end*/hunk.lines).push. /*istanbul ignore start*/apply /*istanbul ignore end*/( /*istanbul ignore start*/_hunk$lines3 /*istanbul ignore end*/, /*istanbul ignore start*/_toConsumableArray( /*istanbul ignore end*/myChanges));
221
      return;
222
    } else if ( /*istanbul ignore start*/(0, _array.arrayStartsWith) /*istanbul ignore end*/(theirChanges, myChanges) && skipRemoveSuperset(mine, theirChanges, theirChanges.length - myChanges.length)) {
0 ignored issues
show
Comprehensibility introduced by
Usage of the sequence operator is discouraged, since it may lead to obfuscated code.

The sequence or comma operator allows the inclusion of multiple expressions where only is permitted. The result of the sequence is the value of the last expression.

This operator is most often used in for statements.

Used in another places it can make code hard to read, especially when people do not realize it even exists as a seperate operator.

This check looks for usage of the sequence operator in locations where it is not necessary and could be replaced by a series of expressions or statements.

var a,b,c;

a = 1, b = 1,  c= 3;

could just as well be written as:

var a,b,c;

a = 1;
b = 1;
c = 3;

To learn more about the sequence operator, please refer to the MDN.

Loading history...
223
      /*istanbul ignore start*/var _hunk$lines4;
224
225
      /*istanbul ignore end*/ /*istanbul ignore start*/(_hunk$lines4 = /*istanbul ignore end*/hunk.lines).push. /*istanbul ignore start*/apply /*istanbul ignore end*/( /*istanbul ignore start*/_hunk$lines4 /*istanbul ignore end*/, /*istanbul ignore start*/_toConsumableArray( /*istanbul ignore end*/theirChanges));
226
      return;
227
    }
228
  } else if ( /*istanbul ignore start*/(0, _array.arrayEqual) /*istanbul ignore end*/(myChanges, theirChanges)) {
0 ignored issues
show
Comprehensibility introduced by
Usage of the sequence operator is discouraged, since it may lead to obfuscated code.

The sequence or comma operator allows the inclusion of multiple expressions where only is permitted. The result of the sequence is the value of the last expression.

This operator is most often used in for statements.

Used in another places it can make code hard to read, especially when people do not realize it even exists as a seperate operator.

This check looks for usage of the sequence operator in locations where it is not necessary and could be replaced by a series of expressions or statements.

var a,b,c;

a = 1, b = 1,  c= 3;

could just as well be written as:

var a,b,c;

a = 1;
b = 1;
c = 3;

To learn more about the sequence operator, please refer to the MDN.

Loading history...
229
    /*istanbul ignore start*/var _hunk$lines5;
230
231
    /*istanbul ignore end*/ /*istanbul ignore start*/(_hunk$lines5 = /*istanbul ignore end*/hunk.lines).push. /*istanbul ignore start*/apply /*istanbul ignore end*/( /*istanbul ignore start*/_hunk$lines5 /*istanbul ignore end*/, /*istanbul ignore start*/_toConsumableArray( /*istanbul ignore end*/myChanges));
232
    return;
233
  }
234
235
  conflict(hunk, myChanges, theirChanges);
236
}
237
238
function removal(hunk, mine, their, swap) {
239
  var myChanges = collectChange(mine),
240
      theirChanges = collectContext(their, myChanges);
241
  if (theirChanges.merged) {
242
    /*istanbul ignore start*/var _hunk$lines6;
243
244
    /*istanbul ignore end*/ /*istanbul ignore start*/(_hunk$lines6 = /*istanbul ignore end*/hunk.lines).push. /*istanbul ignore start*/apply /*istanbul ignore end*/( /*istanbul ignore start*/_hunk$lines6 /*istanbul ignore end*/, /*istanbul ignore start*/_toConsumableArray( /*istanbul ignore end*/theirChanges.merged));
245
  } else {
246
    conflict(hunk, swap ? theirChanges : myChanges, swap ? myChanges : theirChanges);
247
  }
248
}
249
250
function conflict(hunk, mine, their) {
251
  hunk.conflict = true;
252
  hunk.lines.push({
253
    conflict: true,
254
    mine: mine,
255
    theirs: their
256
  });
257
}
258
259
function insertLeading(hunk, insert, their) {
260
  while (insert.offset < their.offset && insert.index < insert.lines.length) {
261
    var line = insert.lines[insert.index++];
262
    hunk.lines.push(line);
263
    insert.offset++;
264
  }
265
}
266
function insertTrailing(hunk, insert) {
267
  while (insert.index < insert.lines.length) {
268
    var line = insert.lines[insert.index++];
269
    hunk.lines.push(line);
270
  }
271
}
272
273 View Code Duplication
function collectChange(state) {
0 ignored issues
show
Duplication introduced by
This code seems to be duplicated in your project.
Loading history...
274
  var ret = [],
275
      operation = state.lines[state.index][0];
276
  while (state.index < state.lines.length) {
277
    var line = state.lines[state.index];
278
279
    // Group additions that are immediately after subtractions and treat them as one "atomic" modify change.
280
    if (operation === '-' && line[0] === '+') {
281
      operation = '+';
282
    }
283
284
    if (operation === line[0]) {
285
      ret.push(line);
286
      state.index++;
287
    } else {
288
      break;
289
    }
290
  }
291
292
  return ret;
293
}
294 View Code Duplication
function collectContext(state, matchChanges) {
0 ignored issues
show
Duplication introduced by
This code seems to be duplicated in your project.
Loading history...
295
  var changes = [],
296
      merged = [],
297
      matchIndex = 0,
298
      contextChanges = false,
299
      conflicted = false;
300
  while (matchIndex < matchChanges.length && state.index < state.lines.length) {
301
    var change = state.lines[state.index],
302
        match = matchChanges[matchIndex];
303
304
    // Once we've hit our add, then we are done
305
    if (match[0] === '+') {
306
      break;
307
    }
308
309
    contextChanges = contextChanges || change[0] !== ' ';
310
311
    merged.push(match);
312
    matchIndex++;
313
314
    // Consume any additions in the other block as a conflict to attempt
315
    // to pull in the remaining context after this
316
    if (change[0] === '+') {
317
      conflicted = true;
318
319
      while (change[0] === '+') {
320
        changes.push(change);
321
        change = state.lines[++state.index];
322
      }
323
    }
324
325
    if (match.substr(1) === change.substr(1)) {
326
      changes.push(change);
327
      state.index++;
328
    } else {
329
      conflicted = true;
330
    }
331
  }
332
333
  if ((matchChanges[matchIndex] || '')[0] === '+' && contextChanges) {
334
    conflicted = true;
335
  }
336
337
  if (conflicted) {
338
    return changes;
339
  }
340
341
  while (matchIndex < matchChanges.length) {
342
    merged.push(matchChanges[matchIndex++]);
343
  }
344
345
  return {
346
    merged: merged,
347
    changes: changes
348
  };
349
}
350
351
function allRemoves(changes) {
352
  return changes.reduce(function (prev, change) {
353
    return prev && change[0] === '-';
354
  }, true);
355
}
356
function skipRemoveSuperset(state, removeChanges, delta) {
357
  for (var i = 0; i < delta; i++) {
358
    var changeContent = removeChanges[removeChanges.length - delta + i].substr(1);
359
    if (state.lines[state.index + i] !== ' ' + changeContent) {
360
      return false;
361
    }
362
  }
363
364
  state.index += delta;
365
  return true;
366
}
367
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/patch/merge.js"],"names":["calcLineCount","merge","hunk","conflicted","oldLines","newLines","lines","forEach","line","mine","theirs","base","loadPatch","ret","index","newFileName","fileNameChanged","oldFileName","oldHeader","newHeader","selectField","hunks","mineIndex","theirsIndex","mineOffset","theirsOffset","length","mineCurrent","oldStart","Infinity","theirsCurrent","hunkBefore","push","cloneHunk","mergedHunk","Math","min","newStart","mergeLines","param","test","Error","undefined","patch","conflict","check","offset","mineLines","theirOffset","theirLines","their","insertLeading","theirCurrent","mutualChange","collectChange","removal","insertTrailing","myChanges","theirChanges","allRemoves","skipRemoveSuperset","swap","collectContext","merged","insert","state","operation","matchChanges","changes","matchIndex","contextChanges","change","match","substr","reduce","prev","removeChanges","delta","i","changeContent"],"mappings":";;;gCAKgBA,a,GAAAA,a;yDA0BAC,K,GAAAA,K;;AA/BhB;;AACA;;AAEA;;;;uBAEO,SAASD,aAAT,CAAuBE,IAAvB,EAA6B;AAClC,MAAIC,aAAa,KAAjB;;AAEAD,OAAKE,QAAL,GAAgB,CAAhB;AACAF,OAAKG,QAAL,GAAgB,CAAhB;;AAEAH,OAAKI,KAAL,CAAWC,OAAX,CAAmB,UAASC,IAAT,EAAe;AAChC,QAAI,OAAOA,IAAP,KAAgB,QAApB,EAA8B;AAC5BL,mBAAa,IAAb;AACA;AACD;;AAED,QAAIK,KAAK,CAAL,MAAY,GAAZ,IAAmBA,KAAK,CAAL,MAAY,GAAnC,EAAwC;AACtCN,WAAKG,QAAL;AACD;AACD,QAAIG,KAAK,CAAL,MAAY,GAAZ,IAAmBA,KAAK,CAAL,MAAY,GAAnC,EAAwC;AACtCN,WAAKE,QAAL;AACD;AACF,GAZD;;AAcA,MAAID,UAAJ,EAAgB;AACd,WAAOD,KAAKE,QAAZ;AACA,WAAOF,KAAKG,QAAZ;AACD;AACF;;AAEM,SAASJ,KAAT,CAAeQ,IAAf,EAAqBC,MAArB,EAA6BC,IAA7B,EAAmC;AACxCF,SAAOG,UAAUH,IAAV,EAAgBE,IAAhB,CAAP;AACAD,WAASE,UAAUF,MAAV,EAAkBC,IAAlB,CAAT;;AAEA,MAAIE,MAAM,EAAV;;AAEA;AACA;AACA;AACA,MAAIJ,KAAKK,KAAL,IAAcJ,OAAOI,KAAzB,EAAgC;AAC9BD,QAAIC,KAAJ,GAAYL,KAAKK,KAAL,IAAcJ,OAAOI,KAAjC;AACD;;AAED,MAAIL,KAAKM,WAAL,IAAoBL,OAAOK,WAA/B,EAA4C;AAC1C,QAAI,CAACC,gBAAgBP,IAAhB,CAAL,EAA4B;AAC1B;AACAI,UAAII,WAAJ,GAAkBP,OAAOO,WAAP,IAAsBR,KAAKQ,WAA7C;AACAJ,UAAIE,WAAJ,GAAkBL,OAAOK,WAAP,IAAsBN,KAAKM,WAA7C;AACAF,UAAIK,SAAJ,GAAgBR,OAAOQ,SAAP,IAAoBT,KAAKS,SAAzC;AACAL,UAAIM,SAAJ,GAAgBT,OAAOS,SAAP,IAAoBV,KAAKU,SAAzC;AACD,KAND,MAMO,IAAI,CAACH,gBAAgBN,MAAhB,CAAL,EAA8B;AACnC;AACAG,UAAII,WAAJ,GAAkBR,KAAKQ,WAAvB;AACAJ,UAAIE,WAAJ,GAAkBN,KAAKM,WAAvB;AACAF,UAAIK,SAAJ,GAAgBT,KAAKS,SAArB;AACAL,UAAIM,SAAJ,GAAgBV,KAAKU,SAArB;AACD,KANM,MAMA;AACL;AACAN,UAAII,WAAJ,GAAkBG,YAAYP,GAAZ,EAAiBJ,KAAKQ,WAAtB,EAAmCP,OAAOO,WAA1C,CAAlB;AACAJ,UAAIE,WAAJ,GAAkBK,YAAYP,GAAZ,EAAiBJ,KAAKM,WAAtB,EAAmCL,OAAOK,WAA1C,CAAlB;AACAF,UAAIK,SAAJ,GAAgBE,YAAYP,GAAZ,EAAiBJ,KAAKS,SAAtB,EAAiCR,OAAOQ,SAAxC,CAAhB;AACAL,UAAIM,SAAJ,GAAgBC,YAAYP,GAAZ,EAAiBJ,KAAKU,SAAtB,EAAiCT,OAAOS,SAAxC,CAAhB;AACD;AACF;;AAEDN,MAAIQ,KAAJ,GAAY,EAAZ;;AAEA,MAAIC,YAAY,CAAhB;AAAA,MACIC,cAAc,CADlB;AAAA,MAEIC,aAAa,CAFjB;AAAA,MAGIC,eAAe,CAHnB;;AAKA,SAAOH,YAAYb,KAAKY,KAAL,CAAWK,MAAvB,IAAiCH,cAAcb,OAAOW,KAAP,CAAaK,MAAnE,EAA2E;AACzE,QAAIC,cAAclB,KAAKY,KAAL,CAAWC,SAAX,KAAyB,EAACM,UAAUC,QAAX,EAA3C;AAAA,QACIC,gBAAgBpB,OAAOW,KAAP,CAAaE,WAAb,KAA6B,EAACK,UAAUC,QAAX,EADjD;;AAGA,QAAIE,WAAWJ,WAAX,EAAwBG,aAAxB,CAAJ,EAA4C;AAC1C;AACAjB,UAAIQ,KAAJ,CAAUW,IAAV,CAAeC,UAAUN,WAAV,EAAuBH,UAAvB,CAAf;AACAF;AACAG,sBAAgBE,YAAYtB,QAAZ,GAAuBsB,YAAYvB,QAAnD;AACD,KALD,MAKO,IAAI2B,WAAWD,aAAX,EAA0BH,WAA1B,CAAJ,EAA4C;AACjD;AACAd,UAAIQ,KAAJ,CAAUW,IAAV,CAAeC,UAAUH,aAAV,EAAyBL,YAAzB,CAAf;AACAF;AACAC,oBAAcM,cAAczB,QAAd,GAAyByB,cAAc1B,QAArD;AACD,KALM,MAKA;AACL;AACA,UAAI8B,aAAa;AACfN,kBAAUO,KAAKC,GAAL,CAAST,YAAYC,QAArB,EAA+BE,cAAcF,QAA7C,CADK;AAEfxB,kBAAU,CAFK;AAGfiC,kBAAUF,KAAKC,GAAL,CAAST,YAAYU,QAAZ,GAAuBb,UAAhC,EAA4CM,cAAcF,QAAd,GAAyBH,YAArE,CAHK;AAIfpB,kBAAU,CAJK;AAKfC,eAAO;AALQ,OAAjB;AAOAgC,iBAAWJ,UAAX,EAAuBP,YAAYC,QAAnC,EAA6CD,YAAYrB,KAAzD,EAAgEwB,cAAcF,QAA9E,EAAwFE,cAAcxB,KAAtG;AACAiB;AACAD;;AAEAT,UAAIQ,KAAJ,CAAUW,IAAV,CAAeE,UAAf;AACD;AACF;;AAED,SAAOrB,GAAP;AACD;;AAED,SAASD,SAAT,CAAmB2B,KAAnB,EAA0B5B,IAA1B,EAAgC;AAC9B,MAAI,OAAO4B,KAAP,KAAiB,QAArB,EAA+B;AAC7B,QAAI,OAAOC,IAAP,CAAYD,KAAZ,KAAuB,WAAWC,IAAX,CAAgBD,KAAhB,CAA3B,EAAoD;AAClD,aAAO,yEAAWA,KAAX,EAAkB,CAAlB;AAAP;AACD;;AAED,QAAI,CAAC5B,IAAL,EAAW;AACT,YAAM,IAAI8B,KAAJ,CAAU,kDAAV,CAAN;AACD;AACD,WAAO,+EAAgBC,SAAhB,EAA2BA,SAA3B,EAAsC/B,IAAtC,EAA4C4B,KAA5C;AAAP;AACD;;AAED,SAAOA,KAAP;AACD;;AAED,SAASvB,eAAT,CAAyB2B,KAAzB,EAAgC;AAC9B,SAAOA,MAAM5B,WAAN,IAAqB4B,MAAM5B,WAAN,KAAsB4B,MAAM1B,WAAxD;AACD;;AAED,SAASG,WAAT,CAAqBN,KAArB,EAA4BL,IAA5B,EAAkCC,MAAlC,EAA0C;AACxC,MAAID,SAASC,MAAb,EAAqB;AACnB,WAAOD,IAAP;AACD,GAFD,MAEO;AACLK,UAAM8B,QAAN,GAAiB,IAAjB;AACA,WAAO,EAACnC,UAAD,EAAOC,cAAP,EAAP;AACD;AACF;;AAED,SAASqB,UAAT,CAAoBS,IAApB,EAA0BK,KAA1B,EAAiC;AAC/B,SAAOL,KAAKZ,QAAL,GAAgBiB,MAAMjB,QAAtB,IACDY,KAAKZ,QAAL,GAAgBY,KAAKpC,QAAtB,GAAkCyC,MAAMjB,QAD7C;AAED;;AAED,SAASK,SAAT,CAAmB/B,IAAnB,EAAyB4C,MAAzB,EAAiC;AAC/B,SAAO;AACLlB,cAAU1B,KAAK0B,QADV,EACoBxB,UAAUF,KAAKE,QADnC;AAELiC,cAAUnC,KAAKmC,QAAL,GAAgBS,MAFrB,EAE6BzC,UAAUH,KAAKG,QAF5C;AAGLC,WAAOJ,KAAKI;AAHP,GAAP;AAKD;;AAED,SAASgC,UAAT,CAAoBpC,IAApB,EAA0BsB,UAA1B,EAAsCuB,SAAtC,EAAiDC,WAAjD,EAA8DC,UAA9D,EAA0E;AACxE;AACA;AACA,MAAIxC,OAAO,EAACqC,QAAQtB,UAAT,EAAqBlB,OAAOyC,SAA5B,EAAuCjC,OAAO,CAA9C,EAAX;AAAA,MACIoC,QAAQ,EAACJ,QAAQE,WAAT,EAAsB1C,OAAO2C,UAA7B,EAAyCnC,OAAO,CAAhD,EADZ;;AAGA;AACAqC,gBAAcjD,IAAd,EAAoBO,IAApB,EAA0ByC,KAA1B;AACAC,gBAAcjD,IAAd,EAAoBgD,KAApB,EAA2BzC,IAA3B;;AAEA;AACA,SAAOA,KAAKK,KAAL,GAAaL,KAAKH,KAAL,CAAWoB,MAAxB,IAAkCwB,MAAMpC,KAAN,GAAcoC,MAAM5C,KAAN,CAAYoB,MAAnE,EAA2E;AACzE,QAAIC,cAAclB,KAAKH,KAAL,CAAWG,KAAKK,KAAhB,CAAlB;AAAA,QACIsC,eAAeF,MAAM5C,KAAN,CAAY4C,MAAMpC,KAAlB,CADnB;;AAGA,QAAI,CAACa,YAAY,CAAZ,MAAmB,GAAnB,IAA0BA,YAAY,CAAZ,MAAmB,GAA9C,MACIyB,aAAa,CAAb,MAAoB,GAApB,IAA2BA,aAAa,CAAb,MAAoB,GADnD,CAAJ,EAC6D;AAC3D;AACAC,mBAAanD,IAAb,EAAmBO,IAAnB,EAAyByC,KAAzB;AACD,KAJD,MAIO,IAAIvB,YAAY,CAAZ,MAAmB,GAAnB,IAA0ByB,aAAa,CAAb,MAAoB,GAAlD,EAAuD;AAAA;;AAAA,8BAC5D;AACA,0EAAK9C,KAAL,EAAW0B,IAAX,4LAAoBsB,cAAc7C,IAAd,CAApB;AACD,KAHM,MAGA,IAAI2C,aAAa,CAAb,MAAoB,GAApB,IAA2BzB,YAAY,CAAZ,MAAmB,GAAlD,EAAuD;AAAA;;AAAA,8BAC5D;AACA,2EAAKrB,KAAL,EAAW0B,IAAX,6LAAoBsB,cAAcJ,KAAd,CAApB;AACD,KAHM,MAGA,IAAIvB,YAAY,CAAZ,MAAmB,GAAnB,IAA0ByB,aAAa,CAAb,MAAoB,GAAlD,EAAuD;AAC5D;AACAG,cAAQrD,IAAR,EAAcO,IAAd,EAAoByC,KAApB;AACD,KAHM,MAGA,IAAIE,aAAa,CAAb,MAAoB,GAApB,IAA2BzB,YAAY,CAAZ,MAAmB,GAAlD,EAAuD;AAC5D;AACA4B,cAAQrD,IAAR,EAAcgD,KAAd,EAAqBzC,IAArB,EAA2B,IAA3B;AACD,KAHM,MAGA,IAAIkB,gBAAgByB,YAApB,EAAkC;AACvC;AACAlD,WAAKI,KAAL,CAAW0B,IAAX,CAAgBL,WAAhB;AACAlB,WAAKK,KAAL;AACAoC,YAAMpC,KAAN;AACD,KALM,MAKA;AACL;AACA8B,eAAS1C,IAAT,EAAeoD,cAAc7C,IAAd,CAAf,EAAoC6C,cAAcJ,KAAd,CAApC;AACD;AACF;;AAED;AACAM,iBAAetD,IAAf,EAAqBO,IAArB;AACA+C,iBAAetD,IAAf,EAAqBgD,KAArB;;AAEAlD,gBAAcE,IAAd;AACD;;AAED,SAASmD,YAAT,CAAsBnD,IAAtB,EAA4BO,IAA5B,EAAkCyC,KAAlC,EAAyC;AACvC,MAAIO,YAAYH,cAAc7C,IAAd,CAAhB;AAAA,MACIiD,eAAeJ,cAAcJ,KAAd,CADnB;;AAGA,MAAIS,WAAWF,SAAX,KAAyBE,WAAWD,YAAX,CAA7B,EAAuD;AACrD;AACA,QAAI,8EAAgBD,SAAhB,EAA2BC,YAA3B,KACGE,mBAAmBV,KAAnB,EAA0BO,SAA1B,EAAqCA,UAAU/B,MAAV,GAAmBgC,aAAahC,MAArE,CADP,EACqF;AAAA;;AAAA,6BACnF,sEAAKpB,KAAL,EAAW0B,IAAX,6LAAoByB,SAApB;AACA;AACD,KAJD,MAIO,IAAI,8EAAgBC,YAAhB,EAA8BD,SAA9B,KACJG,mBAAmBnD,IAAnB,EAAyBiD,YAAzB,EAAuCA,aAAahC,MAAb,GAAsB+B,UAAU/B,MAAvE,CADA,EACgF;AAAA;;AAAA,6BACrF,sEAAKpB,KAAL,EAAW0B,IAAX,6LAAoB0B,YAApB;AACA;AACD;AACF,GAXD,MAWO,IAAI,yEAAWD,SAAX,EAAsBC,YAAtB,CAAJ,EAAyC;AAAA;;AAAA,2BAC9C,sEAAKpD,KAAL,EAAW0B,IAAX,6LAAoByB,SAApB;AACA;AACD;;AAEDb,WAAS1C,IAAT,EAAeuD,SAAf,EAA0BC,YAA1B;AACD;;AAED,SAASH,OAAT,CAAiBrD,IAAjB,EAAuBO,IAAvB,EAA6ByC,KAA7B,EAAoCW,IAApC,EAA0C;AACxC,MAAIJ,YAAYH,cAAc7C,IAAd,CAAhB;AAAA,MACIiD,eAAeI,eAAeZ,KAAf,EAAsBO,SAAtB,CADnB;AAEA,MAAIC,aAAaK,MAAjB,EAAyB;AAAA;;AAAA,2BACvB,sEAAKzD,KAAL,EAAW0B,IAAX,6LAAoB0B,aAAaK,MAAjC;AACD,GAFD,MAEO;AACLnB,aAAS1C,IAAT,EAAe2D,OAAOH,YAAP,GAAsBD,SAArC,EAAgDI,OAAOJ,SAAP,GAAmBC,YAAnE;AACD;AACF;;AAED,SAASd,QAAT,CAAkB1C,IAAlB,EAAwBO,IAAxB,EAA8ByC,KAA9B,EAAqC;AACnChD,OAAK0C,QAAL,GAAgB,IAAhB;AACA1C,OAAKI,KAAL,CAAW0B,IAAX,CAAgB;AACdY,cAAU,IADI;AAEdnC,UAAMA,IAFQ;AAGdC,YAAQwC;AAHM,GAAhB;AAKD;;AAED,SAASC,aAAT,CAAuBjD,IAAvB,EAA6B8D,MAA7B,EAAqCd,KAArC,EAA4C;AAC1C,SAAOc,OAAOlB,MAAP,GAAgBI,MAAMJ,MAAtB,IAAgCkB,OAAOlD,KAAP,GAAekD,OAAO1D,KAAP,CAAaoB,MAAnE,EAA2E;AACzE,QAAIlB,OAAOwD,OAAO1D,KAAP,CAAa0D,OAAOlD,KAAP,EAAb,CAAX;AACAZ,SAAKI,KAAL,CAAW0B,IAAX,CAAgBxB,IAAhB;AACAwD,WAAOlB,MAAP;AACD;AACF;AACD,SAASU,cAAT,CAAwBtD,IAAxB,EAA8B8D,MAA9B,EAAsC;AACpC,SAAOA,OAAOlD,KAAP,GAAekD,OAAO1D,KAAP,CAAaoB,MAAnC,EAA2C;AACzC,QAAIlB,OAAOwD,OAAO1D,KAAP,CAAa0D,OAAOlD,KAAP,EAAb,CAAX;AACAZ,SAAKI,KAAL,CAAW0B,IAAX,CAAgBxB,IAAhB;AACD;AACF;;AAED,SAAS8C,aAAT,CAAuBW,KAAvB,EAA8B;AAC5B,MAAIpD,MAAM,EAAV;AAAA,MACIqD,YAAYD,MAAM3D,KAAN,CAAY2D,MAAMnD,KAAlB,EAAyB,CAAzB,CADhB;AAEA,SAAOmD,MAAMnD,KAAN,GAAcmD,MAAM3D,KAAN,CAAYoB,MAAjC,EAAyC;AACvC,QAAIlB,OAAOyD,MAAM3D,KAAN,CAAY2D,MAAMnD,KAAlB,CAAX;;AAEA;AACA,QAAIoD,cAAc,GAAd,IAAqB1D,KAAK,CAAL,MAAY,GAArC,EAA0C;AACxC0D,kBAAY,GAAZ;AACD;;AAED,QAAIA,cAAc1D,KAAK,CAAL,CAAlB,EAA2B;AACzBK,UAAImB,IAAJ,CAASxB,IAAT;AACAyD,YAAMnD,KAAN;AACD,KAHD,MAGO;AACL;AACD;AACF;;AAED,SAAOD,GAAP;AACD;AACD,SAASiD,cAAT,CAAwBG,KAAxB,EAA+BE,YAA/B,EAA6C;AAC3C,MAAIC,UAAU,EAAd;AAAA,MACIL,SAAS,EADb;AAAA,MAEIM,aAAa,CAFjB;AAAA,MAGIC,iBAAiB,KAHrB;AAAA,MAIInE,aAAa,KAJjB;AAKA,SAAOkE,aAAaF,aAAazC,MAA1B,IACEuC,MAAMnD,KAAN,GAAcmD,MAAM3D,KAAN,CAAYoB,MADnC,EAC2C;AACzC,QAAI6C,SAASN,MAAM3D,KAAN,CAAY2D,MAAMnD,KAAlB,CAAb;AAAA,QACI0D,QAAQL,aAAaE,UAAb,CADZ;;AAGA;AACA,QAAIG,MAAM,CAAN,MAAa,GAAjB,EAAsB;AACpB;AACD;;AAEDF,qBAAiBA,kBAAkBC,OAAO,CAAP,MAAc,GAAjD;;AAEAR,WAAO/B,IAAP,CAAYwC,KAAZ;AACAH;;AAEA;AACA;AACA,QAAIE,OAAO,CAAP,MAAc,GAAlB,EAAuB;AACrBpE,mBAAa,IAAb;;AAEA,aAAOoE,OAAO,CAAP,MAAc,GAArB,EAA0B;AACxBH,gBAAQpC,IAAR,CAAauC,MAAb;AACAA,iBAASN,MAAM3D,KAAN,CAAY,EAAE2D,MAAMnD,KAApB,CAAT;AACD;AACF;;AAED,QAAI0D,MAAMC,MAAN,CAAa,CAAb,MAAoBF,OAAOE,MAAP,CAAc,CAAd,CAAxB,EAA0C;AACxCL,cAAQpC,IAAR,CAAauC,MAAb;AACAN,YAAMnD,KAAN;AACD,KAHD,MAGO;AACLX,mBAAa,IAAb;AACD;AACF;;AAED,MAAI,CAACgE,aAAaE,UAAb,KAA4B,EAA7B,EAAiC,CAAjC,MAAwC,GAAxC,IACGC,cADP,EACuB;AACrBnE,iBAAa,IAAb;AACD;;AAED,MAAIA,UAAJ,EAAgB;AACd,WAAOiE,OAAP;AACD;;AAED,SAAOC,aAAaF,aAAazC,MAAjC,EAAyC;AACvCqC,WAAO/B,IAAP,CAAYmC,aAAaE,YAAb,CAAZ;AACD;;AAED,SAAO;AACLN,kBADK;AAELK;AAFK,GAAP;AAID;;AAED,SAAST,UAAT,CAAoBS,OAApB,EAA6B;AAC3B,SAAOA,QAAQM,MAAR,CAAe,UAASC,IAAT,EAAeJ,MAAf,EAAuB;AAC3C,WAAOI,QAAQJ,OAAO,CAAP,MAAc,GAA7B;AACD,GAFM,EAEJ,IAFI,CAAP;AAGD;AACD,SAASX,kBAAT,CAA4BK,KAA5B,EAAmCW,aAAnC,EAAkDC,KAAlD,EAAyD;AACvD,OAAK,IAAIC,IAAI,CAAb,EAAgBA,IAAID,KAApB,EAA2BC,GAA3B,EAAgC;AAC9B,QAAIC,gBAAgBH,cAAcA,cAAclD,MAAd,GAAuBmD,KAAvB,GAA+BC,CAA7C,EAAgDL,MAAhD,CAAuD,CAAvD,CAApB;AACA,QAAIR,MAAM3D,KAAN,CAAY2D,MAAMnD,KAAN,GAAcgE,CAA1B,MAAiC,MAAMC,aAA3C,EAA0D;AACxD,aAAO,KAAP;AACD;AACF;;AAEDd,QAAMnD,KAAN,IAAe+D,KAAf;AACA,SAAO,IAAP;AACD","file":"merge.js","sourcesContent":["import {structuredPatch} from './create';\nimport {parsePatch} from './parse';\n\nimport {arrayEqual, arrayStartsWith} from '../util/array';\n\nexport function calcLineCount(hunk) {\n  let conflicted = false;\n\n  hunk.oldLines = 0;\n  hunk.newLines = 0;\n\n  hunk.lines.forEach(function(line) {\n    if (typeof line !== 'string') {\n      conflicted = true;\n      return;\n    }\n\n    if (line[0] === '+' || line[0] === ' ') {\n      hunk.newLines++;\n    }\n    if (line[0] === '-' || line[0] === ' ') {\n      hunk.oldLines++;\n    }\n  });\n\n  if (conflicted) {\n    delete hunk.oldLines;\n    delete hunk.newLines;\n  }\n}\n\nexport function merge(mine, theirs, base) {\n  mine = loadPatch(mine, base);\n  theirs = loadPatch(theirs, base);\n\n  let ret = {};\n\n  // For index we just let it pass through as it doesn't have any necessary meaning.\n  // Leaving sanity checks on this to the API consumer that may know more about the\n  // meaning in their own context.\n  if (mine.index || theirs.index) {\n    ret.index = mine.index || theirs.index;\n  }\n\n  if (mine.newFileName || theirs.newFileName) {\n    if (!fileNameChanged(mine)) {\n      // No header or no change in ours, use theirs (and ours if theirs does not exist)\n      ret.oldFileName = theirs.oldFileName || mine.oldFileName;\n      ret.newFileName = theirs.newFileName || mine.newFileName;\n      ret.oldHeader = theirs.oldHeader || mine.oldHeader;\n      ret.newHeader = theirs.newHeader || mine.newHeader;\n    } else if (!fileNameChanged(theirs)) {\n      // No header or no change in theirs, use ours\n      ret.oldFileName = mine.oldFileName;\n      ret.newFileName = mine.newFileName;\n      ret.oldHeader = mine.oldHeader;\n      ret.newHeader = mine.newHeader;\n    } else {\n      // Both changed... figure it out\n      ret.oldFileName = selectField(ret, mine.oldFileName, theirs.oldFileName);\n      ret.newFileName = selectField(ret, mine.newFileName, theirs.newFileName);\n      ret.oldHeader = selectField(ret, mine.oldHeader, theirs.oldHeader);\n      ret.newHeader = selectField(ret, mine.newHeader, theirs.newHeader);\n    }\n  }\n\n  ret.hunks = [];\n\n  let mineIndex = 0,\n      theirsIndex = 0,\n      mineOffset = 0,\n      theirsOffset = 0;\n\n  while (mineIndex < mine.hunks.length || theirsIndex < theirs.hunks.length) {\n    let mineCurrent = mine.hunks[mineIndex] || {oldStart: Infinity},\n        theirsCurrent = theirs.hunks[theirsIndex] || {oldStart: Infinity};\n\n    if (hunkBefore(mineCurrent, theirsCurrent)) {\n      // This patch does not overlap with any of the others, yay.\n      ret.hunks.push(cloneHunk(mineCurrent, mineOffset));\n      mineIndex++;\n      theirsOffset += mineCurrent.newLines - mineCurrent.oldLines;\n    } else if (hunkBefore(theirsCurrent, mineCurrent)) {\n      // This patch does not overlap with any of the others, yay.\n      ret.hunks.push(cloneHunk(theirsCurrent, theirsOffset));\n      theirsIndex++;\n      mineOffset += theirsCurrent.newLines - theirsCurrent.oldLines;\n    } else {\n      // Overlap, merge as best we can\n      let mergedHunk = {\n        oldStart: Math.min(mineCurrent.oldStart, theirsCurrent.oldStart),\n        oldLines: 0,\n        newStart: Math.min(mineCurrent.newStart + mineOffset, theirsCurrent.oldStart + theirsOffset),\n        newLines: 0,\n        lines: []\n      };\n      mergeLines(mergedHunk, mineCurrent.oldStart, mineCurrent.lines, theirsCurrent.oldStart, theirsCurrent.lines);\n      theirsIndex++;\n      mineIndex++;\n\n      ret.hunks.push(mergedHunk);\n    }\n  }\n\n  return ret;\n}\n\nfunction loadPatch(param, base) {\n  if (typeof param === 'string') {\n    if (/^@@/m.test(param) || (/^Index:/m.test(param))) {\n      return parsePatch(param)[0];\n    }\n\n    if (!base) {\n      throw new Error('Must provide a base reference or pass in a patch');\n    }\n    return structuredPatch(undefined, undefined, base, param);\n  }\n\n  return param;\n}\n\nfunction fileNameChanged(patch) {\n  return patch.newFileName && patch.newFileName !== patch.oldFileName;\n}\n\nfunction selectField(index, mine, theirs) {\n  if (mine === theirs) {\n    return mine;\n  } else {\n    index.conflict = true;\n    return {mine, theirs};\n  }\n}\n\nfunction hunkBefore(test, check) {\n  return test.oldStart < check.oldStart\n    && (test.oldStart + test.oldLines) < check.oldStart;\n}\n\nfunction cloneHunk(hunk, offset) {\n  return {\n    oldStart: hunk.oldStart, oldLines: hunk.oldLines,\n    newStart: hunk.newStart + offset, newLines: hunk.newLines,\n    lines: hunk.lines\n  };\n}\n\nfunction mergeLines(hunk, mineOffset, mineLines, theirOffset, theirLines) {\n  // This will generally result in a conflicted hunk, but there are cases where the context\n  // is the only overlap where we can successfully merge the content here.\n  let mine = {offset: mineOffset, lines: mineLines, index: 0},\n      their = {offset: theirOffset, lines: theirLines, index: 0};\n\n  // Handle any leading content\n  insertLeading(hunk, mine, their);\n  insertLeading(hunk, their, mine);\n\n  // Now in the overlap content. Scan through and select the best changes from each.\n  while (mine.index < mine.lines.length && their.index < their.lines.length) {\n    let mineCurrent = mine.lines[mine.index],\n        theirCurrent = their.lines[their.index];\n\n    if ((mineCurrent[0] === '-' || mineCurrent[0] === '+')\n        && (theirCurrent[0] === '-' || theirCurrent[0] === '+')) {\n      // Both modified ...\n      mutualChange(hunk, mine, their);\n    } else if (mineCurrent[0] === '+' && theirCurrent[0] === ' ') {\n      // Mine inserted\n      hunk.lines.push(... collectChange(mine));\n    } else if (theirCurrent[0] === '+' && mineCurrent[0] === ' ') {\n      // Theirs inserted\n      hunk.lines.push(... collectChange(their));\n    } else if (mineCurrent[0] === '-' && theirCurrent[0] === ' ') {\n      // Mine removed or edited\n      removal(hunk, mine, their);\n    } else if (theirCurrent[0] === '-' && mineCurrent[0] === ' ') {\n      // Their removed or edited\n      removal(hunk, their, mine, true);\n    } else if (mineCurrent === theirCurrent) {\n      // Context identity\n      hunk.lines.push(mineCurrent);\n      mine.index++;\n      their.index++;\n    } else {\n      // Context mismatch\n      conflict(hunk, collectChange(mine), collectChange(their));\n    }\n  }\n\n  // Now push anything that may be remaining\n  insertTrailing(hunk, mine);\n  insertTrailing(hunk, their);\n\n  calcLineCount(hunk);\n}\n\nfunction mutualChange(hunk, mine, their) {\n  let myChanges = collectChange(mine),\n      theirChanges = collectChange(their);\n\n  if (allRemoves(myChanges) && allRemoves(theirChanges)) {\n    // Special case for remove changes that are supersets of one another\n    if (arrayStartsWith(myChanges, theirChanges)\n        && skipRemoveSuperset(their, myChanges, myChanges.length - theirChanges.length)) {\n      hunk.lines.push(... myChanges);\n      return;\n    } else if (arrayStartsWith(theirChanges, myChanges)\n        && skipRemoveSuperset(mine, theirChanges, theirChanges.length - myChanges.length)) {\n      hunk.lines.push(... theirChanges);\n      return;\n    }\n  } else if (arrayEqual(myChanges, theirChanges)) {\n    hunk.lines.push(... myChanges);\n    return;\n  }\n\n  conflict(hunk, myChanges, theirChanges);\n}\n\nfunction removal(hunk, mine, their, swap) {\n  let myChanges = collectChange(mine),\n      theirChanges = collectContext(their, myChanges);\n  if (theirChanges.merged) {\n    hunk.lines.push(... theirChanges.merged);\n  } else {\n    conflict(hunk, swap ? theirChanges : myChanges, swap ? myChanges : theirChanges);\n  }\n}\n\nfunction conflict(hunk, mine, their) {\n  hunk.conflict = true;\n  hunk.lines.push({\n    conflict: true,\n    mine: mine,\n    theirs: their\n  });\n}\n\nfunction insertLeading(hunk, insert, their) {\n  while (insert.offset < their.offset && insert.index < insert.lines.length) {\n    let line = insert.lines[insert.index++];\n    hunk.lines.push(line);\n    insert.offset++;\n  }\n}\nfunction insertTrailing(hunk, insert) {\n  while (insert.index < insert.lines.length) {\n    let line = insert.lines[insert.index++];\n    hunk.lines.push(line);\n  }\n}\n\nfunction collectChange(state) {\n  let ret = [],\n      operation = state.lines[state.index][0];\n  while (state.index < state.lines.length) {\n    let line = state.lines[state.index];\n\n    // Group additions that are immediately after subtractions and treat them as one \"atomic\" modify change.\n    if (operation === '-' && line[0] === '+') {\n      operation = '+';\n    }\n\n    if (operation === line[0]) {\n      ret.push(line);\n      state.index++;\n    } else {\n      break;\n    }\n  }\n\n  return ret;\n}\nfunction collectContext(state, matchChanges) {\n  let changes = [],\n      merged = [],\n      matchIndex = 0,\n      contextChanges = false,\n      conflicted = false;\n  while (matchIndex < matchChanges.length\n        && state.index < state.lines.length) {\n    let change = state.lines[state.index],\n        match = matchChanges[matchIndex];\n\n    // Once we've hit our add, then we are done\n    if (match[0] === '+') {\n      break;\n    }\n\n    contextChanges = contextChanges || change[0] !== ' ';\n\n    merged.push(match);\n    matchIndex++;\n\n    // Consume any additions in the other block as a conflict to attempt\n    // to pull in the remaining context after this\n    if (change[0] === '+') {\n      conflicted = true;\n\n      while (change[0] === '+') {\n        changes.push(change);\n        change = state.lines[++state.index];\n      }\n    }\n\n    if (match.substr(1) === change.substr(1)) {\n      changes.push(change);\n      state.index++;\n    } else {\n      conflicted = true;\n    }\n  }\n\n  if ((matchChanges[matchIndex] || '')[0] === '+'\n      && contextChanges) {\n    conflicted = true;\n  }\n\n  if (conflicted) {\n    return changes;\n  }\n\n  while (matchIndex < matchChanges.length) {\n    merged.push(matchChanges[matchIndex++]);\n  }\n\n  return {\n    merged,\n    changes\n  };\n}\n\nfunction allRemoves(changes) {\n  return changes.reduce(function(prev, change) {\n    return prev && change[0] === '-';\n  }, true);\n}\nfunction skipRemoveSuperset(state, removeChanges, delta) {\n  for (let i = 0; i < delta; i++) {\n    let changeContent = removeChanges[removeChanges.length - delta + i].substr(1);\n    if (state.lines[state.index + i] !== ' ' + changeContent) {\n      return false;\n    }\n  }\n\n  state.index += delta;\n  return true;\n}\n"]}
368